热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

放入|时会_性能优化之异步日志

篇首语:本文由编程笔记#小编为大家整理,主要介绍了性能优化之异步日志相关的知识,希望对你有一定的参考价值。     你听说过打印日志能把系统拖垮的情况吗。

篇首语:本文由编程笔记#小编为大家整理,主要介绍了性能优化之异步日志相关的知识,希望对你有一定的参考价值。


          你听说过打印日志能把系统拖垮的情况吗。

          所以项目中一般打印日志会使用异步AsyncAppender打印日志。

          那么使用了AsyncAppender,会不会性能就好了,就不会阻塞业务流程了,会不会丢失日志呢,我们来看一下logback的实现

        

 

                                                                                 

          先看下官方文档的介绍,AsyncAppender 会把处理的事件缓存到一个阻塞队列,默认情况下达到队列容量的80%的时候,会丢弃TRACE, DEBUG and INFO级别的事件,

根据默认配置neverBlock=false,队列也会发生阻塞,所以设置true,虽然完全是非阻塞但会丢失日志,特别是error级别的日志

       我们看下源码如何实现的。

public class AsyncAppender extends AsyncAppenderBase
boolean includeCallerData = false;
/**
* Events of level TRACE, DEBUG and INFO are deemed to be discardable.
* @param event
* @return true if the event is of level TRACE, DEBUG or INFO false otherwise.
*/
protected boolean isDiscardable(ILoggingEvent event)
Level level = event.getLevel();
return level.toInt() <&#61; Level.INFO_INT;

protected void preprocess(ILoggingEvent eventObject)
eventObject.prepareForDeferredProcessing();
if (includeCallerData)
eventObject.getCallerData();

public boolean isIncludeCallerData()
return includeCallerData;

public void setIncludeCallerData(boolean includeCallerData)
this.includeCallerData &#61; includeCallerData;

根据实现&#xff1a;

protected boolean isDiscardable(ILoggingEvent event)
Level level &#61; event.getLevel();
return level.toInt() <&#61; Level.INFO_INT;

public static final int OFF_INT &#61; Integer.MAX_VALUE;
public static final int ERROR_INT &#61; 40000;
public static final int WARN_INT &#61; 30000;
public static final int INFO_INT &#61; 20000;
public static final int DEBUG_INT &#61; 10000;
public static final int TRACE_INT &#61; 5000;
public static final int ALL_INT &#61; Integer.MIN_VALUE;

TRACE, DEBUG and INFO级别的事件&#xff0c;在容量默认到达80%的时候会丢弃。

事件放入阻塞队列的时候即调用put方法时会判断容量及是否可以丢弃条件的判断&#xff0c;如下&#xff1a;

&#64;Override
protected void append(E eventObject)
if (isQueueBelowDiscardingThreshold() && isDiscardable(eventObject))
return;

preprocess(eventObject);
put(eventObject);

private boolean isQueueBelowDiscardingThreshold()
return (blockingQueue.remainingCapacity()
private void put(E eventObject)
if (neverBlock)
blockingQueue.offer(eventObject);
else
putUninterruptibly(eventObject);


private void putUninterruptibly(E eventObject)
boolean interrupted &#61; false;
try
while (true)
try
blockingQueue.put(eventObject);
break;
catch (InterruptedException e)
interrupted &#61; true;


finally
if (interrupted)
Thread.currentThread().interrupt();


当事件真正放入队列&#xff0c;是调用阻塞方法put&#xff0c;还是非阻塞方法offer&#xff0c;是由neverBlock配置决定的&#xff0c;默认false&#xff0c;会阻塞&#xff0c;所以为了性能&#xff0c;我们可以设置为true&#xff0c;也会丢弃日志。

真正处理队列事件的线程是默无闻的后台&#xff08;Daemon&#xff09;消费线程&#xff0c;

Worker worker &#61; new Worker();

worker.setDaemon(true);
worker.setName("AsyncAppender-Worker-" &#43; getName());
// make sure this instance is marked as "started" before staring the worker Thread
super.start();
worker.start()

 

 事件的最终归宿AppenderAttachableImpl aa。

 

class Worker extends Thread
public void run()
AsyncAppenderBase parent &#61; AsyncAppenderBase.this;
AppenderAttachableImpl aai &#61; parent.aai;
// loop while the parent is started
while (parent.isStarted())
try
E e &#61; parent.blockingQueue.take();
aai.appendLoopOnAppenders(e);
catch (InterruptedException ie)
break;


addInfo("Worker thread will flush remaining events before exiting. ");
for (E e : parent.blockingQueue)
aai.appendLoopOnAppenders(e);
parent.blockingQueue.remove(e);

aai.detachAndStopAllAppenders();

 

 

     异步打印日志是否非阻塞&#xff0c;是否会丢弃日志&#xff0c;是完全可以配置的&#xff0c;如果可以丢弃&#xff0c;依据日志的报警系统就不太准了&#xff0c;只能靠打点系统在系统内部aop的方式或其他方式打点上报了。

 

 

 


推荐阅读
  • 本文介绍了深入浅出Linux设备驱动编程的重要性,以及两种加载和删除Linux内核模块的方法。通过一个内核模块的例子,展示了模块的编译和加载过程,并讨论了模块对内核大小的控制。深入理解Linux设备驱动编程对于开发者来说非常重要。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • Java中包装类的设计原因以及操作方法
    本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • JDK源码学习之HashTable(附带面试题)的学习笔记
    本文介绍了JDK源码学习之HashTable(附带面试题)的学习笔记,包括HashTable的定义、数据类型、与HashMap的关系和区别。文章提供了干货,并附带了其他相关主题的学习笔记。 ... [详细]
  • GetWindowLong函数
    今天在看一个代码里头写了GetWindowLong(hwnd,0),我当时就有点费解,靠,上网搜索函数原型说明,死活找不到第 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
author-avatar
张洪幸_246
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有